home
***
CD-ROM
|
disk
|
FTP
|
other
***
search
/
Atari Mega Archive 1
/
Atari Mega Archive - Volume 1.iso
/
gnu
/
gdb
/
gdb_18s.zoo
/
st-infru.c
< prev
next >
Wrap
C/C++ Source or Header
|
1992-03-25
|
34KB
|
1,168 lines
/* this one hacked up for st by jrd */
/* Start and stop the inferior process, for GDB.
Copyright (C) 1986, 1987, 1988 Free Software Foundation, Inc.
GDB is distributed in the hope that it will be useful, but WITHOUT ANY
WARRANTY. No author or distributor accepts responsibility to anyone
for the consequences of using it or for whether it serves any
particular purpose or works at all, unless he says so in writing.
Refer to the GDB General Public License for full details.
Everyone is granted permission to copy, modify and redistribute GDB,
but only under the conditions described in the GDB General Public
License. A copy of this license is supposed to have been given to you
along with GDB so you can know your rights and responsibilities. It
should be in a file named COPYING. Among other things, the copyright
notice and this notice must be preserved on all copies.
In other words, go ahead and share GDB, but don't try to stop
anyone else from sharing it farther. Help stamp out software hoarding!
*/
#include "defs.h"
#include "param.h"
#include "symtab.h"
#include "frame.h"
#include "inferior.h"
#include "wait.h"
#include <stdio.h>
#include <signal.h>
#include <st-out.h>
extern char *sys_siglist[];
extern int errno;
/* Tables of how to react to signals; the user sets them. */
static char signal_stop[NSIG];
static char signal_print[NSIG];
static char signal_program[NSIG];
/* Nonzero if breakpoints are now inserted in the inferior. */
static int breakpoints_inserted;
/* Function inferior was in as of last step command. */
static struct symbol *step_start_function;
/* This is the sequence of bytes we insert for a breakpoint. */
static char break_insn[] = BREAKPOINT;
/* Nonzero => address for special breakpoint for resuming stepping. */
static CORE_ADDR step_resume_break_address;
/* Original contents of the byte where the special breakpoint is. */
static char step_resume_break_shadow[sizeof break_insn];
/* Nonzero means the special breakpoint is a duplicate
so it has not itself been inserted. */
static int step_resume_break_duplicate;
/* Nonzero if we are expecting a trace trap and should proceed from it.
2 means expecting 2 trace traps and should continue both times.
That occurs when we tell sh to exec the program: we will get
a trap after the exec of sh and a second when the program is exec'd. */
static int trap_expected;
/* Nonzero if the next time we try to continue the inferior, it will
step one instruction and generate a spurious trace trap.
This is used to compensate for a bug in HP-UX. */
static int trap_expected_after_continue;
/* Nonzero means expecting a trace trap
and should stop the inferior and return silently when it happens. */
static int stop_after_trap;
/* Nonzero means expecting a trace trap due to attaching to a process. */
static int stop_after_attach;
/* Nonzero if pc has been changed by the debugger
since the inferior stopped. */
int pc_changed;
/* Nonzero if proceed is being used for a "finish" command or a similar
situation when stop_registers should be saved. */
int proceed_to_finish;
/* Nonzero if debugging a remote machine via a serial link or ethernet. */
int remote_debugging;
/* Save register contents here when about to pop a stack dummy frame. */
char stop_registers[REGISTER_BYTES];
/* Nonzero if program stopped due to error trying to insert breakpoints. */
static int breakpoints_failed;
/* Nonzero if inferior is in sh before our program got exec'd. */
static int running_in_shell;
/* Nonzero after stop if current stack frame should be printed. */
static int stop_print_frame;
static void insert_step_breakpoint ();
static void remove_step_breakpoint ();
static void wait_for_inferior ();
static void normal_stop ();
/* Clear out all variables saying what to do when inferior is continued.
First do this, then set the ones you want, then call `proceed'. */
void
clear_proceed_status ()
{
trap_expected = 0;
step_range_start = 0;
step_range_end = 0;
step_frame = 0;
step_over_calls = -1;
step_resume_break_address = 0;
stop_after_trap = 0;
stop_after_attach = 0;
proceed_to_finish = 0;
/* Discard any remaining commands left by breakpoint we had stopped at. */
clear_breakpoint_commands ();
}
/* Basic routine for continuing the program in various fashions.
ADDR is the address to resume at, or -1 for resume where stopped.
SIGNAL is the signal to give it, or 0 for none,
or -1 for act according to how it stopped.
STEP is nonzero if should trap after one instruction.
-1 means return after that and print nothing.
You should probably set various step_... variables
before calling here, if you are stepping.
You should call clear_proceed_status before calling proceed. */
void
proceed (addr, signal, step)
CORE_ADDR addr;
int signal;
int step;
{
int oneproc = 0;
if (step > 0)
step_start_function = find_pc_function (read_pc ());
if (step < 0)
stop_after_trap = 1;
if (addr == -1)
{
/* If there is a breakpoint at the address we will resume at,
step one instruction before inserting breakpoints
so that we do not stop right away. */
if (!pc_changed && breakpoint_here_p (read_pc ()))
oneproc = 1;
}
else
write_register (PC_REGNUM, addr);
if (trap_expected_after_continue)
{
/* If (step == 0), a trap will be automatically generated after
the first instruction is executed. Force step one
instruction to clear this condition. This should not occur
if step is nonzero, but it is harmless in that case. */
oneproc = 1;
trap_expected_after_continue = 0;
}
if (oneproc)
/* We will get a trace trap after one instruction.
Continue it automatically and insert breakpoints then. */
trap_expected = 1;
else
{
int temp = insert_breakpoints ();
if (temp)
{
print_sys_errmsg ("ptrace", temp);
error ("Cannot insert breakpoints.\n\
The same program may be running in another process.");
}
breakpoints_inserted = 1;
}
/* Install inferior's terminal modes. */
terminal_inferior ();
if (signal >= 0)
stop_signal = signal;
/* If this signal should not be seen by program,
give it zero. Used for debugging signals. */
else if (stop_signal < NSIG && !signal_program[stop_signal])
stop_signal= 0;
/* Resume inferior. */
resume (oneproc || step, stop_signal);
/* Wait for it to stop (if not standalone)
and in any case decode why it stopped, and act accordingly. */
wait_for_inferior ();
normal_stop ();
}
/* Writing the inferior pc as a register calls this function
to inform infrun that the pc has been set in the debugger. */
writing_pc (val)
CORE_ADDR val;
{
stop_pc = val;
pc_changed = 1;
}
/* Start an inferior process for the first time.
Actually it was started by the fork that created it,
but it will have stopped one instruction after execing sh.
Here we must get it up to actual execution of the real program. */
start_inferior ()
{
/* We will get a trace trap after one instruction.
Continue it automatically. Eventually (after shell does an exec)
it will get another trace trap. Then insert breakpoints and continue. */
trap_expected = 2;
running_in_shell = 0; /* Set to 1 at first SIGTRAP, 0 at second. */
trap_expected_after_continue = 0;
breakpoints_inserted = 0;
mark_breakpoints_out ();
/* Set up the "saved terminal modes" of the inferior
based on what modes we are starting it with. */
terminal_init_inferior ();
/* Install inferior's terminal modes. */
terminal_inferior ();
if (remote_debugging)
{
trap_expected = 0;
fetch_inferior_registers();
set_current_frame (read_register(FP_REGNUM));
stop_frame = get_current_frame();
inferior_pid = 3;
if (insert_breakpoints())
fatal("Can't insert breakpoints");
breakpoints_inserted = 1;
proceed(-1, -1, 0);
}
else
{
wait_for_inferior ();
normal_stop ();
}
}
/* Start remote-debugging of a machine over a serial link. */
void
start_rem